package com.xy.xymall.order.config;

import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.connection.CorrelationData;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.amqp.support.converter.MessageConverter;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.annotation.PostConstruct;

/**
 * @author XY
 * @version 1.0.0
 * @ClassName MyRabbitConfig.java
 * @createTime 2020年10月31日 11:07:00
 */
@Configuration
public class MyRabbitConfig {

    @Autowired
    RabbitTemplate rabbitTemplate;

    @Bean
    public MessageConverter messageConverter(){
        return new Jackson2JsonMessageConverter();
    }


    /**
     * @author XY
     * @date 2020/11/3 14:32
     * @return void
     * 定制RabbitTemplate
     * 1、服务端收到消息就回调
     *      1、spring.rabbitmq.publisher-confirms=true
     *      2、设置确认回调ConfirmCallback
     * 2、消息正确抵达队列进行回调
     *      1、spring.rabbitmq.publisher-returns=true
     *         spring.rabbitmq.template.mandatory=true
     *      2、设置确认回调ReturnCallback
     * 3、消费端确认（保证每个消息被正确消费，此时才可以broker删除这个消息）
     *      1、默认是自动确认的，只要消息接收到，客户端会自动确认，服务端就会移除这个消息
     *          问题：
     *              我们收到很多消息，自动回复给服务器ack，只有一个消息处理成功，宕机了，发生消息丢失；
     *              手动确认。只要没有明确告诉mq货物被签收。没有ack，消息就一直是unacked状态。即使consumer宕机。消息不会丢失
     *              ，会重新变为Ready，下一次有新的consumer进来就会发给他
     *      2、如何签收：
     *           channel.basicAck(deliveryTag,false);   签收货物
     *           channel.basicNack(deliveryTag,false,true); 拒签
     *
     */
    @PostConstruct  //MyRabbitConfig对象创建完之后会执行这个方法
    public void initRabbitTemplate(){
        //设置确认回调,只要服务端收到消息就会回调
        rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {

            /**
             * @param correlationData 当前消息的唯一关联数据（这个是消息的唯一id）
             * @param ack 消息是否成功收到
             * @param cause 失败的原因
             * @author XY
             * @date 2020/11/3 14:31
             * @return void
             */
            @Override
            public void confirm(CorrelationData correlationData, boolean ack, String cause) {

            }
        });

        //设置消息抵达队列的确认回调
        rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
            /**
             * 只要消息没有投递给指定的队列就触发这个回调
             * @param message 投递失败的消息详细信息
             * @param replyCode 回复的状态码
             * @param replyText 回复的文本内容
             * @param exchange 当时这个消息发给那个交换机
             * @param routingKey 当时这个消息用的路由键
             * @author XY
             * @date 2020/11/3 14:47
             * @return void
             *
             */
            @Override
            public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {

            }
        });

    }


}
